import random
import re
from datetime import datetime

from flask import abort, jsonify,session
from flask import current_app
from flask import make_response
from flask import request

from info import constants, db
from info import redis_store
from info.libt.yuntongxun.sms import CCP
from info.models import User
from info.utils.captcha.response_code import RET
from . import passport_blu
from info.utils.captcha.captcha import captcha


@passport_blu.route('/logout')
def logout():
    """
    退出登陆
    :return:
    """
    # pop是移除session中的数据(dict)
    # pop 会有一个返回值,如果要移除的key不存在,就返回None
    session.pop('mobile', None)
    session.pop('nick_name', None)
    session.pop('user_id', None)
    # 要清楚is_admin的值,如果不清除,先登录管理员,会保存session,再登陆普通用户,又能保存管理员页面
    session.pop('is_admin', None)

    return jsonify(errno=RET.OK, errmsg='退出成功')


@passport_blu.route('/login', methods=['POST'])
def login():
    """
    登陆:
    1.获取参数
    2.校验参数
    3.校验密码是否正确
    4.保存用户的登陆状态
    5.返回响应
    :return:
    """
    # 获取参数
    params_dict = request.json
    mobile = params_dict.get('mobile')
    passport = params_dict.get('password')
    # 校验参数
    if not all([mobile, passport]):
        return jsonify(errno=RET.PARAMERR, errmsg='获取参数失败')

    # 判断手机号是否正确
    if not re.match('1[35678]\\d{9}', mobile):
        return jsonify(errno=RET.PARAMERR, errmsg='参数错误')

    # 校验密码是否正确
    try:
        user = User.query.filter(User.mobile == mobile).first()
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg='数据查询错误')
    # 判断用户是否存在
    if not user:
        return jsonify(errno=RET.NODATA, errmsg='用户不存在')
    # 校验密码是否一致
    if not user.check_password(passport):
        return jsonify(errno=RET.PWDERR, errmsg='用户名或者密码错误')
    # 往session中保存数据表示当前已登陆
    session['user_id'] = user.id
    session['mobile'] = user.mobile
    session['nick_name'] = user.nick_name

    # 设置当前用户最后一次登陆的时间
    # 如果在视图函数中,对模型身上的属性有修改,那么需要commit到数据库保存
    # 但是其实可以不用自己去写 db.session.commit(),前提是对对SQALchemy有过相关配置
    # user.last_login = datetime.now()

    # 返回响应
    return jsonify(errno=RET.OK, errmsg='登陆成功')


@passport_blu.route('/register', methods=['POST'])
def register():
    """
    注册的逻辑:
    1.获取参数
    2.校验参数
    3.取到服务器保存的真实验证码内容
    4.校验用户输入的验证码内容和服务器保存的验证码内容是否一致
    5.如果一致 初始化user模型 添加属性
    6.返回响应
    :return:
    """
    # 1.获取参数
    params_dict = request.json
    mobile = params_dict.get('mobile')
    smscode = params_dict.get('smscode')
    password = params_dict.get('password')

    # 校验参数
    if not all([mobile, smscode, password]):
        return jsonify(errno=RET.PARAMERR, errmsg='参数错误')

    # 取到服务器保存的真实验证码内容
    try:
        real_sms_code = redis_store.get('sms_' + mobile)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg='数据查询失败')
    if not real_sms_code:
        return jsonify(errno=RET.NODATA, errmsg='验证码已过期')
    # 将用户输入的验证码内容和服务器取到的真实验证码内容进行对比
    if smscode != real_sms_code:
        return jsonify(errno=RET.DATAERR, errmsg='验证码输入错误')
    # 初始化user模型 赋值属性
    user = User()
    user.mobile = mobile
    # 暂时没有昵称 用手机号代替
    user.nick_name = mobile
    # 记录用户最后一次登陆时间
    user.last_login = datetime.now()
    # 对密码做处理
    # 在设置password的时候 去对password进行加密 并且将加密结果给user.password_hash进行赋值
    user.password = password

    # 添加到数据库
    try:
        db.session.add(user)
        db.session.commit()
    except Exception as e:
        current_app.logger.error(e)
        db.session.rollback()
        return jsonify(errno=RET.DBERR, errmsg='数据保存失败')
    # 往session中保存数据表示当前已登陆
    session['user_id'] = user.id
    session['mobile'] = user.mobile
    session['nick_name'] = user.nick_name

    # 返回响应
    return jsonify(errno=RET.OK, errmsg='注册成功')


@passport_blu.route('/sms_code', methods=['POST'])
def send_sms_code():
    """
    发送短信的逻辑:
    1. 接收参数并判断是否有值
    2. 校验手机号是正确
    3. 通过传入的图片编码去redis中查询真实的图片验证码内容
    4. 进行验证码内容的比对
    5. 生成发送短信的内容并发送短信
    6. redis中保存短信验证码内容
    7. 返回发送成功的响应
    :return:
    """

    params_dict = request.json
    mobile = params_dict.get('mobile')
    image_code = params_dict.get('image_code')
    image_code_id = params_dict.get('image_code_id')
    # 判断参数是否有值
    if not all([mobile, image_code, image_code_id]):
        return jsonify(errno=RET.PARAMERR, errmsg='参数错误')
    # 判断手机号是否正确
    # if not re.match('1[35678]\\d{9}', mobile):
    #     return jsonify(errno=RET.PARAMERR, errmsg='参数错误')

    # 先从redis取出真实的验证码内容
    try:
        real_image_code = redis_store.get('imageCodeId' + image_code_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg='数据库查询失败')

    if not real_image_code:
        return jsonify(errno=RET.NODATA, errmsg='图片验证码已过期')
    # 与用户的验证码进行对比,如果对比不一致,则返回验证码错误
    if real_image_code.upper() != image_code.upper():
        return jsonify(errno=RET.DATAERR, errmsg='验证码输入错误')
    # 如果一致,生成验证码的内容(随机数据)
    # 随机数字 保证数字长度为6位, 不够在前面补上0
    sms_code_str = '%06d' % random.randint(0, 999999)
    current_app.logger.debug('生成的短信验证码为:%s' % sms_code_str)
    # 发送短信验证码
    # result = CCP().send_template_sms(mobile, [sms_code_str, constants.SMS_CODE_REDIS_EXPIRES / 5], 1)
    # 代表发送短信失败
    # if result != 0:
    #     return jsonify(errno=RET.THIRDERR, errmsg='发送短信失败')
    # 保存验证码到redis
    try:
        redis_store.set('sms_' + mobile, sms_code_str, constants.SMS_CODE_REDIS_EXPIRES)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg='数据保存失败')

    # 发送成功
    return jsonify(errno=RET.OK, errmsg='发送短信成功')


@passport_blu.route('/image_code')
def get_image_code():
    """
    生成图片验证码并返回:
    1.取到参数
    2.判断参数是否有值
    3.生成图片验证码
    4.保存图片验证码的文字到redis
    5.返回验证码图片
    """

    # 1.取到参数 args:取到url中?后面的参数
    image_code_id = request.args.get('imageCodeId', None)
    # 2.判断是否有值
    if not image_code_id:
        return abort(403)
    # 3.生成图片验证码
    name, text, image = captcha.generate_captcha()
    current_app.logger.debug(text)

    # 4.保存图片验证码的文字到redis
    try:
        redis_store.set('imageCodeId' + image_code_id, text, constants.IMAGE_CODE_REDIS_EXPIRES)
    except Exception as e:
        current_app.logger.error(e)
        abort(500)
    # 5.返回验证码图片
    response = make_response(image)
    response.headers['Content-Type'] = 'image/jpg'
    return response
